home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 16 / CU Amiga Magazine's Super CD-ROM 16 (1997-10-16)(EMAP Images)(GB)[!][issue 1997-11].iso / CUCD / Graphics / Ghostscript / source / zstack.c < prev    next >
C/C++ Source or Header  |  1995-01-24  |  7KB  |  263 lines

  1. /* Copyright (C) 1989, 1991, 1992, 1994 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* zstack.c */
  20. /* Operand stack operators */
  21. #include "memory_.h"
  22. #include "ghost.h"
  23. #include "errors.h"
  24. #include "ialloc.h"
  25. #include "istack.h"
  26. #include "oper.h"
  27. #include "store.h"
  28.  
  29. /* <obj> pop - */
  30. int
  31. zpop(register os_ptr op)
  32. {    check_op(1);
  33.     pop(1);
  34.     return 0;
  35. }
  36.  
  37. /* <obj1> <obj2> exch <obj2> <obj1> */
  38. int
  39. zexch(register os_ptr op)
  40. {    ref next;
  41.     check_op(2);
  42.     ref_assign_inline(&next, op - 1);
  43.     ref_assign_inline(op - 1, op);
  44.     ref_assign_inline(op, &next);
  45.     return 0;
  46. }
  47.  
  48. /* <obj> dup <obj> <obj> */
  49. int
  50. zdup(register os_ptr op)
  51. {    check_op(1);
  52.     push(1);
  53.     ref_assign_inline(op, op - 1);
  54.     return 0;
  55. }
  56.  
  57. /* <obj_n> ... <obj_0> <n> index <obj_n> ... <obj_0> <obj_n> */
  58. int
  59. zindex(register os_ptr op)
  60. {    register os_ptr opn;
  61.     check_type(*op, t_integer);
  62.     if ( (ulong)op->value.intval >= op - osbot )
  63.     {    /* Might be in an older stack block. */
  64.         ref *elt;
  65.         if ( op->value.intval < 0 )
  66.           return_error(e_rangecheck);
  67.         elt = ref_stack_index(&o_stack, op->value.intval + 1);
  68.         if ( elt == 0 )
  69.           return_error(e_rangecheck);
  70.         ref_assign(op, elt);
  71.         return 0;
  72.     }
  73.     opn = op + ~(int)op->value.intval;
  74.     ref_assign_inline(op, opn);
  75.     return 0;
  76. }
  77.  
  78. /* <obj_n-1> ... <obj_0> <n> <i> roll */
  79. /*    <obj_(i-1)_mod_ n> ... <obj_0> <obj_n-1> ... <obj_i_mod_n> */
  80. int
  81. zroll(register os_ptr op)
  82. {    os_ptr op1 = op - 1;
  83.     int count, mod;
  84.     register os_ptr from, to;
  85.     register int n;
  86.     check_type(*op1, t_integer);
  87.     check_type(*op, t_integer);
  88.     if ( (ulong)op1->value.intval > op1 - osbot )
  89.     {    /*
  90.          * The data might span multiple stack blocks.
  91.          * There are efficient ways to handle this situation,
  92.          * but they're more complicated than seems worth implementing;
  93.          * for now, do something very simple and inefficient.
  94.          */
  95.         int left, i;
  96.         if ( op1->value.intval < 0 ||
  97.              op1->value.intval + 2 > ref_stack_count(&o_stack)
  98.            )
  99.           return_error(e_rangecheck);
  100.         count = op1->value.intval;
  101.         if ( count <= 1 )
  102.         {    pop(2);
  103.             return 0;
  104.         }
  105.         mod = op->value.intval;
  106.         if ( mod >= count )
  107.             mod %= count;
  108.         else if ( mod < 0 )
  109.         {    mod %= count;
  110.             if ( mod < 0 ) mod += count;  /* can't assume % means mod! */
  111.         }
  112.         /* Use the chain rotation algorithm mentioned below. */
  113.         for ( i = 0, left = count; left; i++ )
  114.           {    ref *elt = ref_stack_index(&o_stack, i + 2);
  115.             ref save;
  116.             int j, k;
  117.             ref *next;
  118.             save = *elt;
  119.             for ( j = i, left--; ; j = k, elt = next, left-- )
  120.               {    k = (j + mod) % count;
  121.                 if ( k == i )
  122.                   break;
  123.                 next = ref_stack_index(&o_stack, k + 2);
  124.                 ref_assign(elt, next);
  125.               }
  126.             *elt = save;
  127.           }
  128.         pop(2);
  129.         return 0;
  130.     }
  131.     count = op1->value.intval;
  132.     if ( count <= 1 )
  133.     {    pop(2);
  134.         return 0;
  135.     }
  136.     mod = op->value.intval;
  137.     /*
  138.      * The elegant approach, requiring no extra space, would be to
  139.      * rotate the elements in chains separated by mod elements.
  140.      * Instead, we simply check to make sure there is enough space
  141.      * above op to do the roll in two block moves.
  142.      * Unfortunately, we can't count on memcpy doing the right thing
  143.      * in *either* direction.
  144.      */
  145.     switch ( mod )
  146.     {
  147.     case 1:            /* common special case */
  148.         pop(2);  op -= 2;
  149.     {    ref top;
  150.         ref_assign_inline(&top, op);
  151.         for ( from = op, n = count; --n; from-- )
  152.             ref_assign_inline(from, from - 1);
  153.         ref_assign_inline(from, &top);
  154.     }    return 0;
  155.     case -1:        /* common special case */
  156.         pop(2);  op -= 2;
  157.     {    ref bot;
  158.         to = op - count + 1;
  159.         ref_assign_inline(&bot, to);
  160.         for ( n = count; --n; to++ )
  161.             ref_assign(to, to + 1);
  162.         ref_assign_inline(to, &bot);
  163.     }    return 0;
  164.     }
  165.     if ( mod < 0 )
  166.       {    mod += count;
  167.         if ( mod < 0 )
  168.           {    mod %= count;
  169.             if ( mod < 0 )
  170.               mod += count;  /* can't assume % means mod! */
  171.           }
  172.       }
  173.     else if ( mod >= count )
  174.       mod %= count;
  175.     if ( mod <= count >> 1)
  176.     {     /* Move everything up, then top elements down. */
  177.         if ( mod >= ostop - op )
  178.           {    o_stack.requested = mod;
  179.             return_error(e_stackoverflow);
  180.           }
  181.         pop(2);  op -= 2;
  182.         for ( to = op + mod, from = op, n = count; n--; to--, from-- )
  183.             ref_assign(to, from);
  184.         memcpy((char *)(from + 1), (char *)(op + 1), mod * sizeof(ref));
  185.     }
  186.     else
  187.     {    /* Move bottom elements up, then everything down. */
  188.         mod = count - mod;
  189.         if ( mod >= ostop - op )
  190.           {    o_stack.requested = mod;
  191.             return_error(e_stackoverflow);
  192.           }
  193.         pop(2);  op -= 2;
  194.         to = op - count + 1;
  195.         memcpy((char *)(op + 1), (char *)to, mod * sizeof(ref));
  196.         for ( from = to + mod, n = count; n--; to++, from++ )
  197.             ref_assign(to, from);
  198.     }
  199.     return 0;
  200. }
  201.  
  202. /* |- ... clear |- */
  203. /* The function name is changed, because the IRIS library has */
  204. /* a function called zclear. */
  205. private int
  206. zclear_stack(os_ptr op)
  207. {    ref_stack_clear(&o_stack);
  208.     return 0;
  209. }
  210.  
  211. /* |- <obj_n-1> ... <obj_0> count <obj_n-1> ... <obj_0> <n> */
  212. private int
  213. zcount(register os_ptr op)
  214. {    push(1);
  215.     make_int(op, ref_stack_count(&o_stack) - 1);
  216.     return 0;
  217. }
  218.  
  219. /* - mark <mark> */
  220. private int
  221. zmark(register os_ptr op)
  222. {    push(1);
  223.     make_mark(op);
  224.     return 0;
  225. }
  226.  
  227. /* <mark> ... cleartomark */
  228. int
  229. zcleartomark(register os_ptr op)
  230. {    uint count = ref_stack_counttomark(&o_stack);
  231.     if ( count == 0 )
  232.       return_error(e_unmatchedmark);
  233.     ref_stack_pop(&o_stack, count);
  234.     return 0;
  235. }
  236.  
  237. /* <mark> <obj_n-1> ... <obj_0> counttomark */
  238. /*    <mark> <obj_n-1> ... <obj_0> <n> */
  239. private int
  240. zcounttomark(os_ptr op)
  241. {    uint count = ref_stack_counttomark(&o_stack);
  242.     if ( count == 0 )
  243.       return_error(e_unmatchedmark);
  244.     push(1);
  245.     make_int(op, count - 1);
  246.     return 0;
  247. }
  248.  
  249. /* ------ Initialization procedure ------ */
  250.  
  251. BEGIN_OP_DEFS(zstack_op_defs) {
  252.     {"0clear", zclear_stack},
  253.     {"0cleartomark", zcleartomark},
  254.     {"0count", zcount},
  255.     {"0counttomark", zcounttomark},
  256.     {"1dup", zdup},
  257.     {"2exch", zexch},
  258.     {"2index", zindex},
  259.     {"0mark", zmark},
  260.     {"1pop", zpop},
  261.     {"2roll", zroll},
  262. END_OP_DEFS(0) }
  263.